三、koa 框架

(一) hello word

  1. 初始化package.json

    新建目录koademo,执行 npm init,一路回车

  2. 创建一个简单的koa应用(hello world)

    安装koa模块 npm install koa --save 在项目根目录新建app.js,app.js代码如下:

const Koa = require('koa');
const app = new Koa();

app.use(async ctx => {
  ctx.body = 'Hello World';
});

app.listen(3000);
  1. 执行node app.js
  2. 用浏览器访问 http://localhost:3000

(二) 服务器自动重新部署

我们发现没修改一次代码,都必须重启才能生效,这显然不方便,我们希望修改了文件服务能自动重启.nodemon能实现这样功能

  1. 全局安装npm i nodemon -g
  2. 启动服务的时候用nodemon app.js 代替node app.js

(三) koa路由 koa-router

  1. 创建新文件夹koa-app作为项目的跟目录

  2. 使用终端(命令提示符或者powershell)进入根目录,初始化package.json

    npm init -y
    
  3. 安装koa和koa-router

    npm i koa koa-router --save-dev
    
  4. 新建app.js,代码如下

    const Koa = require('koa')
    const Router = require('koa-router')
    const app = new Koa()
    const router = new Router();
    
    router.get('/', ctx => {
      ctx.body = 'hello world'
    })
    router.get('/demo1', ctx => {
      ctx.body = 'demo1'
    })
    router.post('/demo1', ctx => {
      ctx.body = 'demo1'
    })
    router.all('/demo2', ctx => {
      ctx.body = 'demo2'
    })
    
    app.use(router.routes());
    
    app.listen(3000, () => {
      console.log('服务已启动,在 http://localhost:3000/');
    });
    
  5. 启动服务,用浏览器访接口即可

  6. 封装router,我们一般会把router单独放到一个文件,然后倒入到app.js使用

    const Router = require('koa-router')
    const router = new Router();
    
    router.all('/', ctx => {
      ctx.body = 'hello world'
    })
    router.all('/add', ctx => {
      ctx.body = 'add'
    })
    router.all('/getList', ctx => {
      ctx.body = 'demo1'
    })
    router.all('/del', ctx => {
      ctx.body = 'del'
    })
    router.all('/edit', ctx => {
      ctx.body = 'edit'
    })
    
    module.exports = router;
    

    app.js里的代码变成这样

    const Koa = require('koa')
    const app = new Koa()
    const router = require('./router');
    
    app.use(router.routes());
    
    app.listen(3000, () => {
      console.log('服务已启动,在 http://localhost:3000/');
    });
    
  7. 配置父路由

    const Router = require('koa-router');
    
    const router = new Router({
      // 配置前缀
      prefix: '/city'
    })
    
    router.get('/add',(ctx,next)=> {
    	ctx.body = '增加';
    })
    
    router.get('/del',ctx=> {
    	ctx.body = '删除';
    })
    router.get('/edit',ctx=> {
    	ctx.body = '编辑';
    })
    router.get('/list',ctx=> {
    	ctx.body = '查询';
    })
      
    module.exports = router;
    

    浏览器访问http://localhost:3000/city/add(或其他) 即可

(四) koa中间件

一个请求从发出到返回数据,如果我们想在这个过程里做一些统一的操作,可以使用中间件来完成.

  1. 中间件是个函数

  2. 使用中间件的方式, app.use(中间件)

    app.use((ctx, next) => {
    	// 在ctx上放入username,后面的所有请求的ctx里都会有username这个变量
      ctx.username = '我是老胡';
      // 处理完之后放行,不使用next()的话,程序会被挂起来不动了
      next();  // 在vue路由守卫里曾使用过next
    })
    
  3. 中间件有顺序

  4. 中间的运行顺序

    koa å›¾ç‰‡ä¸Šä¼ è¯¦è§£

(五) 设置静态目录

  1. 在目录中创建目录static,在public下创建文件demo.html,访问http://localhost:3000/public/demo.html是无法访问得到,因为我们还没有设置静态资源目录,设置静态资源目录要用到koa-static模块

  2. 安装koa-static

    npm i koa-static --save-dev
    
  3. 在app.j是里加入如下代码

const koaStatic = require('koa-static');
app.use(koaStatic(__dirname + '/public'));

再来访问 http://localhost:3000/demo.html 就可以访问了,ps: 路径不用加public

(六) 获取请求参数

给刚才的demo.html添加axios用来发送请求,代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>Document</title>
  <script src="https://cdn.bootcss.com/axios/0.19.0-beta.1/axios.js"></script>
</head>
<body>
  <h3>
    测试koa静态资源目录
  </h3>

  <script>
    let url = 'http://localhost:3000/add';
    let data = {
      params: {
        username: 'laohu',
        phone: 13800000000
      }
    }
    axios.get(url, data).then(res => {
      console.log(res);
    })

  </script>
</body>
</html>

  1. 获取get请求参数 在router.js的add接口里加入如下代码
router.all('/add', ctx => {
  // 获取get请求参数
  const username = ctx.query.username;
  const phone = ctx.query.phone;
  // 把拿到的数据放入ctx.body
  ctx.body = {
    module: 'add',
    username,
    phone
  }
})
  1. 获取post请求 获取post请求需要使用koa-body模块 安装koa-body npm i koa-body --save

    npm i koa-body --save
    

    在app.js里加入如下代码:

const koaBody = require('koa-body');
app.use(koaBody());

获取post请求参数的代码如下

ctx.request.body.xxx
  1. 使用中间件封装请求参数

    // 把get请求参数和post请求参数都放入params对象
    app.use((ctx, next) => {
      ctx.params = {
        ...ctx.query,
        ...ctx.request.body
      }
      next();
    });
    

    07 使用模板(了解)

一般请求一个接口返回的是一坨数据,然而有时候我们希望返回的是一个html网页或者一段html代码(上周分享的服务器渲染) 我们试用koa-swig模块来向前端返回一个html

  1. 安装koa-swig
npm i koa-view --save-dev

在根目录创建views目录,在views目录下创建tpl.html,代码如下

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <title>Document</title>
</head>

<body>
  <h3><%=title %></h3>
  <ul>
    <%
      list.forEach(item=>{
    %>
    <li><%=item.username %> => <%=item.age %></li>
    <%
    })
    %>
  </ul>
</body>

</html>
  1. 在app.js添加如下代码:
const views = require("koa-views");
app.use(views(path.join(__dirname, "views"), {
  map: {
    html: 'underscore'
  }
}));
   
// 在路由中使用   
router.all('/', async ctx => {
  const list = [{
      username: '约书亚',
      age: 22
    },
    {
      username: '亚伯拉罕',
      age: 100
    },
    {
      username: '挪亚',
      age: 500
    }
  ]

  await ctx.render('tpl', {
    title: 'kao-框架实战',
    list
  });
})
  1. 访问 http://localhost:3000,就可以看到一个html页面

ps: 要访问一个接口,吐出一个页面你还可以直接在js里使用反引号``拼接模板,当然也可以使用vue来渲染

(七) vue服务器渲染

  1. 安装vue和vue服务器渲染插件

    npm install vue vue-server-renderer --save
    
  2. 创建渲染vue实例

    // 第 1 步:创建一个 Vue 实例
    const Vue = require('vue')
    const app = new Vue({
      template: `<div>Hello World</div>`
    })
    // 第 2 步:创建一个 renderer
    const renderer = require('vue-server-renderer').createRenderer();
    
    // 第 3 步:将 Vue 实例渲染为 HTML
    renderer.renderToString(app, (err, html) => {
      if (err) throw err
      console.log(html)
      // => <div data-server-rendered="true">Hello World</div>
    })
    
    

(八) 跨域配置

自己编写一个中间件即可,代码如下

app.use((ctx, next) => {
  ctx.set("Access-Control-Allow-Origin", "*");
  ctx.set("Access-Control-Allow-Methods", "PUT, POST, GET, DELETE, OPTIONS");
  // 请求头设置
  ctx.set(
    "Access-Control-Allow-Headers",
    `Content-Type, Content-Length, Authorization, Accept, X-Requested-With , yourHeaderFeild,x-token,sessionToken,token`
  );
  if (ctx.method == "OPTIONS") {
    ctx.body = 200;
  } else {
    next();
  }
})

访问刚才的demo.html文件, http://localhost:3000/demo.html,你会发现你的请求的响应头上添加了Access-Control-Allow-Origin: *,说明服务器跨域设置已经成功了.

(九) koa-body实现文件的上传和下载

  1. 安装koa-body中间件

  2. 在app.js使用koa-body中间件

    app.use(koaBody({
      formLimit: '1mb',
      multipart: true, // 允许上传多个文件
      formidable:{
        maxFieldsSize:200*1024*1024,  // 文件大小
        keepExtensions:true // 保存图片的扩展名
      }
    }));
    
  3. 编写接口

      async one(ctx) {
        try {
          const file = ctx.request.files.file;
          let fileName = file.name;
          // 创建可读流
          const render = fs.createReadStream(file.path);
          // 指定存放路径
          let filePath = path.join(__dirname, '../public/upload/',fileName);
          const upStream = fs.createWriteStream(filePath);
          render.pipe(upStream);
          // 给前端返回图片地址,xxx.xxx.xxx:xxxx是你的服务器地址和端口号
          let imgUrl = 'http://xxx.xxx.xxx:xxxx/'+'/upload/'+fileName;
          ctx.body = {
            code: 666,
            msg: '上传成功',
            result: {
              imgUrl
            }
          }
        } catch (error) {
          ctx.body = {
          	code: 500,
          	msg: '上传失败'
          }
        }
      }
    

参考链接:

https://blog.csdn.net/meifannao789456/article/details/88662840

(十) nodejs搭建https服务器

  1. 生成证书和密钥
  2. 把存放证书和密钥重名为 certificate, 拷贝到koa应用的根目录
  3. 编写代码 app.js
//使用nodejs自带的http、https模块
const https = require('https');
const http = require('http');
const fs = require('fs');
const path = require('path');
const koa = require('koa');
const app = new koa();

app.on('error', (error, ctx) => {
    console.log('something error ' + JSON.stringify(ctx.onerror));
});

app.use(async ctx => {
    ctx.body = `This is ${ctx.protocol} visit`;
});

//根据项目的路径导入生成的证书文件
const privateKey  = fs.readFileSync(path.join(__dirname, './certificate/server.key'), 'utf8');
const certificate = fs.readFileSync(path.join(__dirname, './certificate/server.crt'), 'utf8');
const credentials = {key: privateKey, cert: certificate};

//创建http与HTTPS服务器
const httpServer = http.createServer(app.callback());
const httpsServer = https.createServer(credentials, app.callback());

//可以分别设置http、https的访问端口号
const PORT = 8000;
const SSLPORT = 8001;

//创建http服务器
httpServer.listen(PORT, function() {
    console.log('HTTP Server is running on: http://localhost:%s', PORT);
});

//创建https服务器
httpsServer.listen(SSLPORT, function() {
    console.log('HTTPS Server is running on: https://localhost:%s', SSLPORT);
});

参考链接: https://www.jianshu.com/p/4d4ae558deff

(十一) koa2爬虫

参考地址:

Nodejs -- 使用koa2搭建数据爬虫

一只node爬虫的升级打怪之路

(十二) koa中使用ts

  1. Package.json

    {
      "name": "myapp",
      "version": "1.0.0",
      "description": "",
      "main": "",
      "scripts": {
        "start": "ts-node-dev app.ts"
      },
      "author": "nero",
      "license": "ISC",
      "devDependencies": {
        "@types/koa": "^2.11.3",
        "@types/node": "^14.0.12",
        "koa": "^2.12.0",
        "ts-node": "^8.10.2",
        "typescript": "^3.9.5"
      },
      "dependencies": {
        "ts-node-dev": "^1.1.8"
      }
    }
    
    
  2. app.ts

    import * as Koa from "koa";
    
    const app = new Koa();
    
    app.use(async ctx => {
        ctx.body = 'Hello World';
    });
    
    //设置监听端口
    app.listen(3000, () => {
        console.log("服务器开启 http://127.0.0.1:3000");
    });
    

(十三) 图片上传

app.js

const koaBody = require("koa-body");

 app.use(koaBody({
  multipart: true,
  formidable: {
    //上传文件存储目录
    uploadDir: path.join(__dirname, `/public/upload/`),
    //允许保留后缀名
    keepExtensions: true,
    multipart: true,
  },
  jsonLimit: '10mb',
  formLimit: '10mb',
  textLimit: '10mb'
})); //解析formdata过来的数据 

uploadRouter.js

router.all('/upload/one', ctx => {
    let imgUrl = 'http://' + ctx.request.header.host + '/upload' + ctx.request.files.file.path.split("\\upload\\")[1];
    ctx.body = {
        code: 666,
        msg: 'success',
        data: {
            imgUrl
        }
    };
});

前端文件-html

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>

<body>
    <input type="file">
    <button onclick="upload()">upload</button>

    <script src="https://cdn.bootcdn.net/ajax/libs/axios/0.27.2/axios.min.js"></script>
    <script>
        function upload() {
            var _input = document.querySelector('input');
            var file = _input.files[0];
            var formData = new FormData();
            formData.append('file', file);
            axios({
                method: "post",
                headers: {
                    "Content-Type": "multipart/form-data",
                },
                url: "/upload",
                data: formData,
            })
                .then(function (response) {
                    if (response.data.code == "200") {
                        console.log(response);
                    }
                })
                .catch(function (error) {
                    console.log(error);
                });
        }


    </script>
</body>

</html>

前端文件-vue

<template>
    <div id="app">
        <h3>单个图片上传</h3>
        <el-upload
            class="avatar-uploader"
            action="https://jsonplaceholder.typicode.com/posts/"
            :http-request="uploadRequest"
            :before-upload="beforeUpload"
        >
            <img v-if="imageUrl" :src="imageUrl" class="avatar" />
            <i v-else class="el-icon-plus avatar-uploader-icon"></i>
        </el-upload>

        <h3>多个图片上传</h3>
        <el-upload
            class="avatar-uploader"
            action="#"
            multiple
            :auto-upload="false"
            :on-change="onChange"
        >
            <template v-if="imageUrlList.length>0">
                <img  :src="item" class="avatar" v-for="(item,index) in imageUrlList" :key="index" />
            </template>
            <i v-else class="el-icon-plus avatar-uploader-icon"></i>
        </el-upload>
        <el-button @click="uploadRequestMultiple" type="primary">点击上传</el-button>
    </div>
</template>

<script>
import { upload, multipleUpload } from './api/index'
export default {
    name: 'App',
    data () {
        return {
            imageUrl: '',
            file: '',
            files: [],
            imageUrlList: []
        }
    },
    methods: {
        beforeUpload (file, fileList) {
            this.file = fileList
        },
        onChange (file, fileList) {
            this.files = fileList
        },
        uploadRequestMultiple () {
            const formdata = new FormData()
            this.files.forEach(item => {
                formdata.append('avatar', item.raw)
            })
            multipleUpload(formdata).then(res => {
                this.imageUrlList = res.data.url
            })
        },
        uploadRequest () {
            const formdata = new FormData()
            formdata.append('avatar', this.file)
            upload(formdata).then(res => {
                this.imageUrl = res.data.url
            })
        }
    }
}
</script>

<style>
.avatar-uploader .el-upload {
  border: 1px dashed #d9d9d9;
  border-radius: 6px;
  cursor: pointer;
  position: relative;
  overflow: hidden;
}
.avatar-uploader .el-upload:hover {
  border-color: #409eff;
}
.avatar-uploader-icon {
  font-size: 28px;
  color: #8c939d;
  width: 178px;
  height: 178px;
  line-height: 178px;
  text-align: center;
}
.avatar {
  width: 178px;
  height: 178px;
  display: block;
}
</style>

(十四) websocket

(1) 简单版本

app.js

// 导入WebSocket模块:
const WebSocket = require('ws'); // npm i ws

// 引用Server类:
const WebSocketServer = WebSocket.Server;

// 实例化:
const wss = new WebSocketServer({
    port: 3000
});   

wss.on('connection', function (ws) {
  ws.on('message', function (message) {
      ws.send(2222); 
  })
});

浏览器端

<body>
  <script>
    // 打开一个WebSocket:
    var ws = new WebSocket('ws://localhost:3000');
    // 响应onmessage事件:
    ws.onmessage = function (msg) { console.log(msg); };
    ws.onopen = function () {
      // 给服务器发送一个字符串:
      ws.send('Hello!');
    }

  </script>
</body>

(2) 进阶版

git@gitee.com:huruqing/socket.git